**React Testing Library(RTL)**는 React 컴포넌트를 사용자 관점에서 테스트하기 위해 만들어진 라이브러리입니다.
npm install --save-dev @testing-library/react @testing-library/jest-dom
@testing-library/react
: 핵심 기능 제공 (렌더링, 쿼리)@testing-library/jest-dom
: Jest에서 사용할 수 있는 DOM 전용 Matcher 제공// Counter.js
import { useState } from "react";
export default function Counter() {
const [count, setCount] = useState(0);
return (
<div>
<p>Count: {count}</p>
<button onClick={() => setCount(count + 1)}>증가</button>
</div>
);
}
// Counter.test.js
import { render, screen } from "@testing-library/react";
import userEvent from "@testing-library/user-event";
import Counter from "./Counter";
test("버튼 클릭 시 카운트가 증가한다", async () => {
render(<Counter />);
const button = screen.getByRole("button", { name: "증가" });
await userEvent.click(button);
expect(screen.getByText("Count: 1")).toBeInTheDocument();
});
RTL은 요소를 찾을 때 접근성(Accessibility) 기준을 권장합니다.
getByRole
(가장 권장)getByLabelText
getByText
getByPlaceholderText
getByTestId
(최후의 수단)screen.getByRole("button", { name: "로그인" });
screen.getByLabelText("아이디");
screen.getByPlaceholderText("비밀번호 입력");
screen.getByText("회원가입");
메서드 | 요소 없을 때 | 사용 시점 |
---|---|---|
getBy* | 에러 발생 | 반드시 존재해야 할 때 |
queryBy* | null 반환 | 없을 수도 있을 때 |
findBy* | Promise 반환 | 비동기 요소 탐색 시 |
userEvent
유틸을 통해 실제 사용자의 상호작용을 시뮬레이션합니다.const input = screen.getByRole("textbox");
await userEvent.type(input, "Hello");
expect(input).toHaveValue("Hello");
@testing-library/jest-dom
을 설치하면 DOM 전용 Matcher를 활용할 수 있습니다.
toBeInTheDocument()
: 요소 존재 여부toHaveTextContent()
: 텍스트 포함 여부toBeVisible()
: 보이는지 여부toHaveValue()
: 입력 값 검증expect(screen.getByText("로그인")).toBeInTheDocument();
expect(screen.getByRole("textbox")).toHaveValue("Hello");
사용자 관점(User-centric) → 버튼을 "클래스명"이 아니라 역할(Role) 과 레이블(Label) 로 찾자.
구현 세부사항(Implementation detail)을 피하자 → state 값 직접 접근하지 말고, UI를 통해 검증하자.
접근성(Accessibility)을 중시하자 → 스크린리더 등 실제 사용자 환경을 고려한 테스트 작성.